home *** CD-ROM | disk | FTP | other *** search
/ MacWorld 1999 March - Disc 1 / Macworld (1999-03) (Disk 1).dmg / Shareware World / Utilities / Text Processing / Alpha / Tcl / Packages / Docprojects.tcl < prev    next >
Encoding:
Text File  |  1998-12-16  |  32.6 KB  |  1,031 lines  |  [TEXT/ALFA]

  1. ## -*-Tcl-*-
  2.  # ###################################################################
  3.  #  Vince's Additions - an extension package for Alpha
  4.  # 
  5.  #  FILE: "Docprojects.tcl"
  6.  #                                    created: 29/7/97 {4:59:22 pm} 
  7.  #                                last update: 16/12/1998 {1:42:07 pm} 
  8.  #  Author: Vince Darley
  9.  #  E-mail: <darley@fas.harvard.edu>
  10.  #    mail: Division of Engineering and Applied Sciences, Harvard University
  11.  #          Oxford Street, Cambridge MA 02138, USA
  12.  #     www: <http://www.fas.harvard.edu/~darley/>
  13.  #  
  14.  # Copyright (c) 1997-1998  Vince Darley, all rights reserved
  15.  # 
  16.  # See the file "license.terms" for information on usage and redistribution
  17.  # of this file, and for a DISCLAIMER OF ALL WARRANTIES.
  18.  # ###################################################################
  19.  ##
  20.  
  21. alpha::extension documentProjects 1.6.2 {
  22.     set alpha::prefs(documentProjects) Docproj
  23.     alpha::package require -loose Alpha 7.1p3
  24.     namespace eval Docproj {}
  25.     # dummy value
  26.     ensureset docProject(name) [list "None" "Project2" "Thesis"]
  27.     # The name of the current project.  Every project has a unique name
  28.     newPref var currentProject "None" Docproj "" docProject(name) "varitem"
  29.     # Different identities can be useful if your projects may be sometimes
  30.     # for work purposes, sometimes for your own purposes etc.
  31.     newPref var identity Usual Docproj Docproj::changeIdentity identities "array"
  32.     menu::buildProc "Current Project" \
  33.       {menu::buildFlagMenu "Current Project" list currentProject DocprojmodeVars}
  34.     menu::insert packagePrefs submenu 0 {Current Project}
  35.     menu::insert packagePrefs items 0 \
  36.       "documentProjectPrefs…" "userDetails…" \
  37.       "<E<SremoveDocumentTemplate…" "<S<BeditDocumentTemplate…" \
  38.       "<SnewDocumentTemplate…" \
  39.       "<E<SremoveProject…" "<S<BeditProject…" "<SnewProject…"
  40.     # Key-binding to update the version number in a file's header.
  41.     # These version numbers can be inserted by some of the standard
  42.     # document templates.
  43.     newPref binding updateFileVersion "/f<U" Docproj
  44.     menu::insert winUtils items end \
  45.         "updateDate" \
  46.         "[menu::bind DocprojmodeVars(updateFileVersion) -]"
  47.     lunion elec::MenuTemplates "createHeader" "newDocument"
  48.     catch "unBind F1 bind::Completion"
  49.     menu::insert elec items end \
  50.         {Menu -n FunctionComments -p menu::generalProc {
  51.         "/eusual"    
  52.         "/e<Isimple" 
  53.         "/e<OwithAuthor" 
  54.         "/e<Uupdate" 
  55.     }}
  56.     namespace eval newDocument {}
  57.     set "newDocument::handlers(Document Projects)" Docproj::newHandler
  58.     # Use this simple proc if we don't have the newDocument package.
  59.     if {![alpha::package exists newDocument]} {
  60.         ;proc file::newDocument {} {
  61.             beep
  62.             Docproj::newHandler [list -n [statusPrompt "New doc name:"]]
  63.         }
  64.     } else {
  65.         alpha::package require newDocument
  66.     }
  67.     
  68.     # When you request a new document, if this flag is set the user
  69.     # is only prompted with a list of document templates which 
  70.     # are relevant to the current mode.  This can be useful if you 
  71.     # have lots of templates.
  72.     newPref flag docTemplatesModeSpecific 1 Docproj
  73.     # When a file is saved, its header (time-stamp) etc can be
  74.     # automatically updated.
  75.     newPref flag autoUpdateHeader 1 Docproj
  76.     # call on saveHook
  77.     proc Docproj::changeProject {name} {
  78.         if {$name == "*"} { return }
  79.         menu::flagProc "Current Project" $name
  80.     }
  81.     
  82.     # call on saveHook
  83.     hook::register saveHook updateHeaderHook
  84. } maintainer {
  85.     "Vince Darley" darley@fas.harvard.edu <http://www.fas.harvard.edu/~darley/>
  86. } uninstall {this-file} help {file "Documentprojects Help"}
  87.  
  88. # user projects
  89. if ![info exists docProject(addendum)] {
  90.     set docProject(addendum) { {none} {about some other stuff} {deep problems}}
  91.     set docProject(default_modes) { {} {C++ Tcl} {TeX}}
  92.     set docProject(extra) [list "" "Freely distributable" "Copyright (C) 1997-1998 the author."]
  93.     set docProject(license) [list "" "" ""]
  94. }
  95.  
  96.  
  97. proc updateHeaderHook {name} {
  98.     global DocprojmodeVars
  99.     if $DocprojmodeVars(autoUpdateHeader) {
  100.     # update does no harm if it fails so we call it for all
  101.     # modes with no worries.
  102.     getWinInfo -w $name a
  103.     if {$a(dirty)} {
  104.         file::updateDate $name
  105.     }
  106.     }
  107. }
  108.  
  109. # header/source templates (NOTE: FORMAT OF THIS LIST MAY CHANGE)
  110. llunion elec::DocTemplates 1 \
  111.     { * "Empty" * "" *} \
  112.     { * "Default" * t_default *} \
  113.     { TeX "Basic LaTeX document" "None" t_latex * {article report letter book slides}} \
  114.     { C++ "Basic C++ header file" "Header" t_cpp_header * } \
  115.     { C++ "Basic C++ source file" "Source" t_cpp_source * } \
  116.     { HTML "HTML document" * t_html * } 
  117.     ## 
  118.      # \
  119.      # { C++ "Cpptcl Class Source" Source t_cpptcl_source "Cpptcl"} \
  120.      # { C++ "Cpptcl Class Header" Header t_cpptcl_header "Cpptcl"} \
  121.      # { Tcl "Itcl Class" * t_itcl_class "Cpptcl"}  \
  122.      # { Tcl "Blank Tcl Header" Header "\#" "Vince's Additions"} \
  123.      # { C++ "EvoX Class Source" Source t_cpptcl_source "EvoX"} \
  124.      # { C++ "EvoX Class Header" Header t_cpptcl_header "EvoX"}
  125.      ##
  126.  
  127. # used for file description headers
  128. if $synchroniseWithInternetConfig {
  129.     catch {set user(author) [icGetPref RealName]}
  130.     catch {set user(email) "<[icGetPref Email]>"}
  131.     catch {set user(www) "<[icGetPref WWWHomePage]>"}
  132.     catch {set user(organisation) [icGetPref Organization]}
  133. ensureset user(author) "Ken McKen"
  134. ensureset user(email) "ken@kenny.com"
  135. ensureset user(www) "http://www.kenny.com/"
  136. ensureset user(organisation) "Ken Corp."
  137.  
  138. ensureset user(address) "Rose St, MA 02143, USA"
  139. ensureset user(author_initials) "VMD"
  140.  
  141. ensureset identities(Usual) [array get user]
  142.  
  143. proc Docproj::changeIdentity {var} {
  144.     global identities user DocprojmodeVars
  145.     array set user $identities($DocprojmodeVars($var))
  146. }
  147.  
  148. if {[info exists DocprojmodeVars(identity)]} {
  149.     Docproj::changeIdentity identity
  150. }
  151.  
  152. ## 
  153.  # ###################################################################
  154.  #    Used to be "docProjEngine.tcl", now one file:
  155.  # ###################################################################
  156.  ##
  157.  
  158. proc global::userDetails {} {
  159.     global DocprojmodeVars modifiedArrayElements user identities
  160.     dialog::pkg_options "Docprojects" \
  161.       "User Details (some may be from Internet Config)" 1 user
  162.     if {![dialog::yesno -y "Update" -n "New Identity" \
  163.       "Update $DocprojmodeVars(identity) identity, or make a new one?"]} {
  164.     # Ask for new name
  165.     set name [eval prompt [list "Enter tag for new identity" \
  166.       "<Tag>" "Old ids:"] [array names identities]]
  167.     set identities($name) [array get user]
  168.     set DocprojmodeVars(identity) $name
  169.     # Have to store Usual id too.
  170.     lappend modifiedArrayElements [list $name identities] \
  171.       [list identity DocprojmodeVars] [list Usual identities]
  172.     }
  173. }
  174.  
  175. proc global::documentProjectPrefs {} {
  176.     dialog::pkg_options "Docproj" "Preferences for your Document Projects"
  177. }
  178.  
  179. proc Docproj::newHandler {args} {
  180.     set doc [file::createDocument "new $args"]
  181.     if {[getModifiers] & 72} {
  182.     file::pickProject
  183.     }
  184.     file::createHeader $doc
  185.     return ""
  186. }
  187.  
  188. proc file::pickProject {} {
  189.     global DocprojmodeVars docProject
  190.     set item [listpick -p "Pick a project…" -L $DocprojmodeVars(currentProject) \
  191.       $docProject(name)]
  192.     if {$item != ""} {Docproj::changeProject $item }
  193.     return $item
  194. }
  195.  
  196. proc file::projectName {} { 
  197.     global DocprojmodeVars
  198.     return $DocprojmodeVars(currentProject)
  199. }
  200.  
  201. proc file::projectAddendum {} {
  202.     global docProject DocprojmodeVars
  203.     return [lindex $docProject(addendum) \
  204.       [lsearch -exact $docProject(name) $DocprojmodeVars(currentProject)]]
  205. }
  206.  
  207. proc file::projectExtra {} {
  208.     global docProject DocprojmodeVars
  209.     return [lindex $docProject(extra) \
  210.       [lsearch -exact $docProject(name) $DocprojmodeVars(currentProject)]]
  211. }
  212. proc file::projectLicense {} {
  213.     global docProject DocprojmodeVars
  214.     set ret [lindex $docProject(license) \
  215.       [lsearch -exact $docProject(name) $DocprojmodeVars(currentProject)]]
  216.     if {$ret == ""} {
  217.     return "none"
  218.     } else {
  219.     return $ret
  220.     }
  221. }
  222.  
  223. namespace eval functioncomments {}
  224.  
  225. ## 
  226.  # ----------------------------------------------------------------------
  227.  #     
  228.  #    "file::functionComment" --
  229.  #    
  230.  #     This procedure    generates a    nice little    comment    box
  231.  #     like this one here.
  232.  #    
  233.  #    Results:
  234.  #     Well it doesn't return    anything, but it allows    you    to
  235.  #     enter each    item simply, moving    from one to the next with Tab
  236.  #    
  237.  #    Side effects:
  238.  #     Not much
  239.  #    
  240.  # ----------------------------------------------------------------------
  241.  ##
  242. proc functioncomments::usual { {simple ""} {author 0} } {
  243.     global user
  244.     set fn [getSelect]
  245.     set fn [lindex $fn end]
  246.     beginningOfLine
  247.     set t "-------------------------------------------------------------------------\r"
  248.     append t "\r"
  249.     append t "\"$fn\" --\r"
  250.     append t "\r •description•\r"
  251.     if { $simple != "simple" } {
  252.     append t "\rResults:\r •results•\r\rSide effects:\r •side effects•\r"
  253.     }
  254.     if $author {
  255.     append t "\r--Version--Author------------------Changes-------------------------------"
  256.     append t "\r   1.0     $user(email) original\r"
  257.     }
  258.     append t "-------------------------------------------------------------------------"
  259.     set t [file::commentTextBlock $t]
  260.     elec::CenterInsertion $t
  261. }
  262.  
  263. proc functioncomments::simple {} { return [functioncomments::usual simple 0]}
  264. proc functioncomments::withAuthor {} { return [functioncomments::usual "" 1] }
  265.  
  266. proc file::commentTextBlock {text} {
  267.     set cc [commentCharacters "Paragraph"]
  268.     set c [lindex $cc 2]
  269.     regsub -all "\[\r\n\]" $text "\r${c}" text
  270.     return "[lindex $cc 0]\r[lindex $cc 2]${text}\r[lindex $cc 1]\r"
  271. }
  272.  
  273. ## 
  274.  # -------------------------------------------------------------------------
  275.  #     
  276.  #    "file::functionCommentUpdate" --
  277.  #    
  278.  #     Handles updating of a version line    like the one below
  279.  #    
  280.  # --Version--Author------------------Changes-------------------------------  
  281.  #      1.0      <darley@fas.harvard.edu> original
  282.  #    1.1     <darley@fas.harvard.edu> quickly updated with shift-F1
  283.  # -------------------------------------------------------------------------
  284.  ##
  285. proc functioncomments::update {} {
  286.     global user
  287.     set begin [lindex [commentCharacters Paragraph] 2]
  288.     goto [file::findLocally "${begin}--Version--Author"]
  289.     goto [nextLineStart [nextLineStart [getPos] ]]
  290.     goto [file::findLocally "${begin}-------"]
  291.     elec::Insertion "${begin}   •Version•     $user(email) •Changes•\r"
  292. }
  293.  
  294. ## 
  295.  # -------------------------------------------------------------------------
  296.  #     
  297.  #    "file::findLocally" --
  298.  #    
  299.  #     Looks around for a    particular sequence    of characters (or a    regexp)
  300.  #     and returns the start of the closest fit, either fowards or backwards,
  301.  #     or    "" if no match was found.
  302.  # -------------------------------------------------------------------------
  303.  ##
  304. proc file::findLocally { chars {regexp 0} { pos "" } } {
  305.     if { $pos == "" } { set pos [getPos] }
  306.     
  307.     set found1 [lindex [search -s -f 0 -n -r $regexp -- "$chars" $pos] 0]
  308.     set found2 [lindex [search -s -f 1 -n -r $regexp -- "$chars" $pos] 0]
  309.     
  310.     if { $found1 != "" && $found2 != "" } { 
  311.     if [expr ([pos::math $pos + 0] - [pos::math $found1 + 0]) \
  312.       <= ([pos::math $found2 + 0] - [pos::math $pos + 0]) ] {
  313.         return $found1
  314.     } else {
  315.         return $found2
  316.     }
  317.     }
  318.     
  319.     # return whatever we can, possibly ""
  320.     if { $found1 != "" } {
  321.     return $found1
  322.     } else {
  323.     if { $found2 == "" } { 
  324.         message "Couldn't find: $chars"
  325.     }
  326.     return $found2
  327.     }
  328. }
  329.  
  330.  
  331. ## 
  332.  # -------------------------------------------------------------------------
  333.  #     
  334.  #    "file::updateFileVersion"    --
  335.  #    
  336.  #     Update    the    version    number and information in the header block
  337.  #     of    a file.  Copes with both my old and new formats.
  338.  #    
  339.  # -------------------------------------------------------------------------
  340.  ##
  341. proc file::updateFileVersion {} {
  342.     global user
  343.     # in case the user wishes to return quickly
  344.     pushPosition
  345.     
  346.     goto [minPos]
  347.     set begin [lindex [commentCharacters Paragraph] 2]
  348.     set pos [file::findLocally "_/_/_" 0]
  349.     if { $pos == "" || [pos::compare $pos > [pos::math [minPos] + 1000]]} {
  350.     set srch [quote::WhitespaceReg [quote::Regfind "${begin} " ]]
  351.     append srch {[0-9]+/[0-9]+/[0-9]+}
  352.     set pos [file::findLocally $srch 1]
  353.     if { $pos == "" } {
  354.         message "Couldn't find original version template."
  355.         set srch [quote::Regfind "${begin} "]
  356.         append srch "See header file for further information"
  357.         set pos [file::findLocally [quote::WhitespaceReg $srch]]
  358.         if { $pos != "" } {
  359.         set pos [nextLineStart $pos]
  360.         } else {
  361.         goto [minPos]
  362.         set pos [file::findLocally "${begin}\#\#\#"]
  363.         if { $pos == "" } { message "Couldn't find any header" ; return }
  364.         set pos [lindex [search -s -f 1 -n -- "${begin}\#\#\#" [nextLineStart $pos]] 0]
  365.         if { $pos == "" } { message "Couldn't find any header" ; return }
  366.         }
  367.         goto $pos
  368.         set t  "${begin}\r"
  369.         append t  "${begin} modified by  rev reason\r"
  370.         append t  "${begin} ---------- --- --- -----------\r"
  371.         append t  "${begin} [file::paddedDate] $user(author_initials) 1.0 original\r"
  372.         insertText $t
  373.         select $pos [getPos]
  374.         return ""
  375.     } else {
  376.         # This is the normal case.
  377.         # Find the last version number
  378.         set p [minPos]
  379.         while {[pos::compare $p != $pos]} {
  380.         set pos $p
  381.         set p [file::findLocally $srch 1 [nextLineStart $p] ]
  382.         }
  383.         set pos [nextLineStart $pos]
  384.     }    
  385.     } else {
  386.     # old style header
  387.     set pos [lineStart $pos]
  388.     replaceText $pos [nextLineStart $pos] ""
  389.     }
  390.     # Now pos is at the start of the line where we wish to insert
  391.     goto $pos
  392.     elec::Insertion "${begin} [file::paddedDate] $user(author_initials) •• ••\r"
  393.     message "Pop position to return to where you were."
  394.     return ""
  395. }
  396.  
  397. proc file::paddedDate {{when ""}} {
  398.     if {$when == ""} { set when [now] }
  399.     return [string range "[lindex [mtime $when short] 0]     " 0 9]
  400. }
  401.  
  402. proc file::created {{convert 1}} {
  403.     if [catch {getFileInfo [win::Current] info}] {
  404.     if $convert {
  405.         return [mtime [now]]
  406.     } else {
  407.         return [now]
  408.     }
  409.     } else {
  410.     if $convert {
  411.         return [mtime $info(created)]
  412.     } else {
  413.         return $info(created)
  414.     }
  415.     }        
  416. }
  417.  
  418.  
  419. ## 
  420.  # -------------------------------------------------------------------------
  421.  #     
  422.  #    "file::createHeader" --
  423.  #    
  424.  #     Insert    a descriptive header into the current file.     Needs to be tailored 
  425.  #     more to different modes, but isn't    too    bad    right now.
  426.  #     
  427.  #     'forcemode' will force    the    file into that mode    via    emacs-like mode
  428.  #     entries on    the    top    line of    the    file. 
  429.  #     
  430.  #     'parent' gives    the    name of    a class    from which the generated file
  431.  #     descends (appropriate for C++,    [incr Tcl] for example).
  432.  # 
  433.  # -------------------------------------------------------------------------
  434.  ##
  435. proc file::createHeader { {template ""} {parent "" } } {
  436.     # Make sure the current project is compatible with this mode
  437.     file::coordinateProjectForMode
  438.     if {$parent == ""} {set parent "•parent•"}
  439.     if {$template == ""} { set template [list "" "" "Header" "\#" "" ""] }
  440.     # make the header
  441.     if {[lindex $template 1] != "Empty" } {
  442.     set t ""
  443.     set class [file::className]
  444.     if {$class == "Untitled"} {set class "•class name•"}
  445.     set file [win::CurrentTail]
  446.     set docHeadType [lindex $template 2]
  447.     if {$docHeadType != "None" } {
  448.         append t [file::topHeader]
  449.         if {$docHeadType != "Basic"} {
  450.         if {$docHeadType == "Source" || [file::isSource $file]} {
  451.             # it's a source file
  452.             append t " See header file for further information\r"
  453.         } elseif {$docHeadType == "Header" || $docHeadType == "*" && [file::isHeader $file]} {
  454.             global user
  455.             append t " Description: \r"
  456.             append t "\r"
  457.             append t " History\r"
  458.             append t "\r"
  459.             append t " modified by  rev reason\r"
  460.             append t " ---------- --- --- -----------\r"
  461.             append t " [file::paddedDate [file::created 0]] $user(author_initials) 1.0 original\r"
  462.         } else {
  463.             # not header or source or basic... oh well!
  464.         }
  465.         }
  466.         append t "###################################################################"
  467.         set t [file::commentTextBlock $t]
  468.         global mode
  469.         global ${mode}::firstHeaderLine
  470.         if {[info exists ${mode}::firstHeaderLine]} {
  471.         regsub "\r" $t "[quote::Regsub [set ${mode}::firstHeaderLine]]\r" t                
  472.         } else {
  473.         regsub "\r" $t "-*-${mode}-*-\r" t
  474.         }
  475.     }
  476.     set procName [lindex $template 3]
  477.     if {$procName != "\#" && [info commands $procName] == ""} { 
  478.         global PREFS
  479.         if [catch {uplevel \#0 source [list [file join $PREFS prefs.tcl]]}] {
  480.         alertnote "An error occurred while loading \"prefs.tcl\"" 
  481.         global errorInfo
  482.         dumpTraces "prefs.tcl error" $errorInfo
  483.         error ""
  484.         }            
  485.     }
  486.     if [catch {append t [eval $procName [list $class] [list $parent] [lindex $template 5]]}] {
  487.         alertnote "An error occurred while calling \"$procName\"" 
  488.         global errorInfo
  489.         dumpTraces "$procName error" $errorInfo
  490.         error ""
  491.     }
  492.     goto [minPos]
  493.     elec::Insertion $t
  494.     }
  495.     return ""
  496. }
  497.  
  498. ## 
  499.  # -------------------------------------------------------------------------
  500.  #     
  501.  #    "file::createDocument" --
  502.  #    
  503.  #     Make a new document from a given template type.
  504.  #     
  505.  #     'forcemode' will force    the    file into that mode    via    emacs-like mode
  506.  #     entries on    the    top    line of    the    file. 
  507.  #     
  508.  # -------------------------------------------------------------------------
  509.  ##
  510. proc file::createDocument { {winCreate ""} {forcemode "" } } {
  511.     # pick a template
  512.     # if [fileIsHeader    $file]
  513.     global elec::DocTemplates mode DocprojmodeVars
  514.     # decide if its mode-specific or not
  515.     set f [lindex $winCreate 2]
  516.     if $DocprojmodeVars(docTemplatesModeSpecific) {
  517.     if {$forcemode != ""} {
  518.         set tlist [file::docTemplates $f $forcemode non]
  519.     } else {
  520.         set tlist [file::docTemplates $f $mode non]
  521.     }
  522.     } else {
  523.     set tlist [file::docTemplates $f "" non]
  524.     }
  525.     lappend tlist "<Create new document type>"
  526.     if {$non != ""} {
  527.     eval lappend tlist "----------------------------------------------------" [lsort $non]
  528.     }
  529.     set tchoice [listpick -p "Pick a document template to insert" -L "Default" $tlist]
  530.     if {$tchoice == "<Create new document type>"} {
  531.     set tchoice [file::newDocumentTemplate 1]
  532.     }
  533.     if {$tchoice == "----------------------------------------------------"} { error "" }
  534.     
  535.     set tinfo [file::docTemplateInfo $tchoice]
  536.     set subTypes [lindex $tinfo 5]
  537.     if {$subTypes != ""} {
  538.     # replace the list of options with just the one selected
  539.     set tinfo [lreplace $tinfo 5 5 [listpick -p "Pick a document subtype of $tchoice" $subTypes]]
  540.     }
  541.     if {$forcemode == "" && [lindex $tinfo 0] != "*"} {
  542.     set forcemode [lindex $tinfo 0]
  543.     }
  544.     if {$winCreate != ""} {
  545.     eval $winCreate
  546.     }
  547.     
  548.     if { $forcemode != "" && $mode != $forcemode} { 
  549.     changeMode $forcemode
  550.     }
  551.     # we need to do this to stop modes switching later if this file isn't
  552.     # obviously a '$mode' file.
  553.     global win::Modes
  554.     set win::Modes($f) $mode
  555.     # set the project
  556.     Docproj::changeProject [lindex $tinfo 4]
  557.     # if the current project doesn't like this mode, then switch
  558.     file::coordinateProjectForMode
  559.     return $tinfo
  560. }
  561.  
  562. proc file::docTemplates { {f ""} {modeSpecific ""} {other ""}} {
  563.     global elec::DocTemplates
  564.     if {$other != ""} { upvar $other noList }
  565.     set tlist ""
  566.     set noList ""
  567.     if {$f != "" && $f != "Untitled"} {
  568.     set m [file::whichModeForWin $f]
  569.     foreach t ${elec::DocTemplates} {
  570.         if [file::docTemplateMatchExt $t $f $m] {
  571.         lappend tlist [lindex $t 1]
  572.         } else {
  573.         lappend noList [lindex $t 1]
  574.         }
  575.     }        
  576.     } else {
  577.     foreach t ${elec::DocTemplates} {
  578.         if {$modeSpecific == "" || [string match [lindex $t 0] $modeSpecific]} {
  579.         lappend tlist [lindex $t 1]
  580.         } else {
  581.         lappend noList [lindex $t 1]
  582.         }
  583.     }        
  584.     }    
  585.     return [lsort $tlist]
  586. }
  587.  
  588. proc file::docTemplateMatchExt {t f {m ""}} {
  589.     if {$m == ""} {set m [file::whichModeForWin $f]}
  590.     # match everything to a file with no particular extension
  591.     if {$m == "Text"} { return 1 }
  592.     set l [lindex $t 0]
  593.     set mMatch [expr [lsearch -exact $l $m] != -1]
  594.     switch [lindex $t 2] {
  595.     "None" -
  596.     "Basic" -
  597.     "*" {
  598.         if {$l == "*"} {
  599.         return 1
  600.         } else {
  601.         return $mMatch
  602.         }
  603.     }
  604.     "Header" {
  605.         if {$mMatch} {
  606.         return [file::isHeader $f $m]
  607.         }
  608.     }
  609.     "Source" {
  610.         if {$mMatch} {
  611.         return [file::isSource $f $m]
  612.         }
  613.         
  614.     }
  615.     }
  616.     return 0
  617. }
  618.  
  619. proc file::docTemplateInfo {name} {
  620.     global elec::DocTemplates
  621.     foreach t ${elec::DocTemplates} {
  622.     if {$name == [lindex $t 1]} {
  623.         return $t
  624.     }
  625.     }
  626. }
  627. proc file::docTemplateIndex {name} {
  628.     set i 0
  629.     global elec::DocTemplates
  630.     foreach t ${elec::DocTemplates} {
  631.     if {$name == [lindex $t 1]} {
  632.         return $i
  633.     }
  634.     incr i
  635.     }
  636. }
  637.  
  638. proc file::notTextMode {} {
  639.     global mode mode::features
  640.     if { $mode == "Text" } {
  641.     # we probably don't want Text mode     
  642.     set m [listpick -p "Pick a mode:" -L "Text" [array names mode::features]]
  643.     if { $m == "" } {set m "Text"}
  644.     changeMode $m
  645.     } 
  646. }
  647.  
  648. ## 
  649.  # -------------------------------------------------------------------------
  650.  #     
  651.  #    "file::topHeader"    --
  652.  #    
  653.  #     Inserts the top part of a    descriptive    header into    the    current    file
  654.  # -------------------------------------------------------------------------
  655.  ##
  656. proc file::topHeader { } {
  657.     global user
  658.     set file [win::CurrentTail]
  659.     if [catch {getFileInfo [win::Current] info}] {
  660.     set created [mtime [now]]
  661.     set last_update $created
  662.     } else {
  663.     set created [mtime $info(created)]
  664.     set last_update [mtime $info(modified)]
  665.     }        
  666.     append t "###################################################################\r"
  667.     if {[file::projectName] != "*"} {
  668.     append t " [file::projectName] - [file::projectAddendum]\r"
  669.     }
  670.     append t "\r" 
  671.     append t " FILE: \"" $file "\"\r"
  672.     append t "                                   created: $created \r"
  673.     append t "                               last update: $last_update \r"    
  674.     append t " Author: $user(author)\r"
  675.     append t " E-mail: $user(email)\r"
  676.     if {$user(organisation) != ""} {
  677.     append t "   mail: $user(organisation)\r"
  678.     }
  679.     if {$user(address) != ""} {
  680.     append t "         $user(address)\r"
  681.     }
  682.     if {$user(www) != ""} {
  683.     append t "    www: $user(www)\r"
  684.     }
  685.     append t " \r"
  686.     append t [file::[file::projectLicense]]
  687.     if {[set e [file::projectExtra]] != ""} {
  688.     append t "[breakIntoLines $e]\r \r"
  689.     }
  690.     return $t
  691. }
  692.  
  693. ## 
  694.  # -------------------------------------------------------------------------
  695.  #     
  696.  #    "file::className"    --
  697.  #    
  698.  #     Extract root of file name as a    class name for the file    (obviously most    
  699.  #     relevant to C++)  
  700.  # -------------------------------------------------------------------------
  701.  ##
  702. proc file::className {} { return [file::baseName [win::CurrentTail]] }
  703.  
  704.  
  705. ## 
  706.  # -------------------------------------------------------------------------
  707.  #   
  708.  #  "file::coordinateProjectForMode" --
  709.  #  
  710.  #   When we create a new file or header automatically, it contains
  711.  #   information about our current project (as defined in docProject(...)).
  712.  #   Unfortunately we often forget to select the correct project first.
  713.  #   This procedure makes sure that your project is compatible with the
  714.  #   current mode, given the information in the 'docProject' array. If it isn't
  715.  #   then the current project is changed if a better match can be found. 
  716.  #         
  717.  #  Results:
  718.  #   None
  719.  #  
  720.  #  Side effects:
  721.  #   The current project may be changed
  722.  # -------------------------------------------------------------------------
  723.  ##
  724. proc file::coordinateProjectForMode {} {
  725.     global mode docProject
  726.     set currProj [file::projectName]
  727.     set projModes [lindex $docProject(default_modes) \
  728.       [lsearch -exact $docProject(name) [file::projectName]]]
  729.     if { $projModes != "" && [lsearch -exact $projModes $mode] == -1 } {
  730.     # this project doesn't like this mode.
  731.     # see if there's a better one
  732.     foreach modeLists $docProject(default_modes) {
  733.         if { [lsearch -exact $modeLists $mode] != -1 } {
  734.         # found a fit
  735.         set index [lsearch -exact $docProject(default_modes) $modeLists]
  736.         set proj [lindex $docProject(name) $index]
  737.         Docproj::changeProject "$proj"
  738.         return
  739.         }
  740.     }
  741.     }
  742. }
  743.  
  744. proc file::createNewClass {} {
  745.     global mode
  746.     # if the current project doesn't like this mode, then switch
  747.     file::coordinateProjectForMode
  748.     beep
  749.     set class [statusPrompt "A name for the new class:"]
  750.     set parent [statusPrompt "Descended from:" ]
  751.     switch $mode {
  752.     "C" -
  753.     "C++" {
  754.         file::createHeader [file::createDocument "new -n ${class}.cc" C++] $parent
  755.         file::createHeader [file::createDocument "new -n ${class}.h" C++] $parent
  756.     } 
  757.     "Tcl" {
  758.         file::createHeader [file::createDocument "new -n ${class}.tcl" Tcl] $parent
  759.     }
  760.     default {
  761.         message "No class procedure defined for your mode. Why not write one yourself?"
  762.     }
  763.     
  764.     }            
  765.     
  766. }
  767.  
  768.  
  769. ## 
  770.  # -------------------------------------------------------------------------
  771.  #   
  772.  # "file::updateGeneralDate" --
  773.  #  
  774.  #  Updates the date in the header of a file.  Normally this is the 
  775.  #  'last update' date, but we can override that if desired.
  776.  # -------------------------------------------------------------------------
  777.  ##
  778. proc file::updateGeneralDate { name {patt ""} {time ""}} {
  779.     if {$patt == ""} {set patt {last update: }}
  780.     regsub -all { } $patt "\[ \t\]" spatt
  781.     set pos [getPos]
  782.     set end [selEnd]
  783.     set hour {[0-9][0-9]?(:|\.)[0-9][0-9]((:|\.)[0-9][0-9])?([ \t][APap][Mm])?}
  784.     set date {[0-9][0-9]?(/|\.|\-)[0-9][0-9]?(/|\.|\-)[0-9][0-9]([0-9][0-9])?}
  785.     append spatt "\[ \t\]*" $date "\[ \t]\{?" $hour {\}?}
  786.     set datePos [search -s -n -f 1 -r 1 -m 0 -l [pos::math [minPos] + 1000] $spatt [minPos]]
  787.     if {![llength $datePos]} {return}
  788.     if {$time == ""} {set time [mtime [now] short]}
  789.     if {[eval getText $datePos] == $time} {return}
  790.     eval replaceText $datePos [list $patt $time]
  791.     select $pos $end
  792.     return
  793. }
  794.  
  795. proc file::updateDate { {name ""} } {
  796.     set fr [win::Current]
  797.     if { $name == "" } {
  798.     set name $fr
  799.     }
  800.     if { $name != $fr } {
  801.     bringToFront $name
  802.     file::updateGeneralDate $name
  803.     bringToFront $fr
  804.     } else {
  805.     file::updateGeneralDate $name
  806.     }    
  807. }
  808.  
  809. proc file::updateCreationDate { name } {
  810.     if [catch {getFileInfo [stripNameCount [win::Current]] info}] {
  811.     set created [mtime [now]]
  812.     } else {
  813.     set created [mtime $info(created)]
  814.     }        
  815.     file::updateGeneralDate $name "created" $created
  816. }
  817.  
  818. proc file::newFunction {} {
  819.     elec::Insertion "[file::className]::•name•(•args•){\r\t•body•\r}\r"
  820. }
  821.  
  822. proc global::newDocumentTemplate { {subCall 0} } {
  823.     set newT [global::_editDocumentTemplate]
  824.     global elec::DocTemplates 
  825.     lappend elec::DocTemplates $newT
  826.     # save it permanently
  827.     global modifiedVars
  828.     lappend modifiedVars elec::DocTemplates
  829.     # add template to "prefs.tcl"
  830.     set procedure [lindex $newT 3]
  831.     set subproj [lindex $newT 5]
  832.     if {$procedure != "\#"} {
  833.     set def [file::_getDefault "Do you want to use this as the template?" t]
  834.     set t "\r"
  835.     append t "proc $procedure \{docname parentdoc"
  836.     if {$subproj != ""} { append t " subtype " }
  837.     append t "\} \{\r"
  838.     append t "\t# You must fill this in\r"
  839.     if {$subproj != ""} { append t "\t# Possible 'subtypes' are: $subproj\r" }
  840.     append t $def
  841.     append t "\r\treturn \$t\r\}\r"
  842.     addUserLine $t
  843.     if {[askyesno "I've added a template for the procedure to your 'prefs.tcl'. Do you want to edit it now?"] == "yes"} {
  844.         global::editPrefsFile
  845.         goto [maxPos]
  846.         if $subCall { 
  847.         alertnote "Once you've finished editing, hit cmd-N to go back and create a new document." 
  848.         # so our calling proc stops
  849.         error "Editing"
  850.         }
  851.     }
  852.     }
  853.     return [lindex $newT 1]
  854. }
  855.  
  856. proc file::_varValue {var} {
  857.     upvar $var a
  858.     if [info exists a] {
  859.     return $a
  860.     } else {
  861.     return ""
  862.     }
  863. }
  864.  
  865. proc file::_getDefault { text {default ""} {var ""}} {
  866.     if {[isSelection]} {
  867.     if {[askyesno "I notice you've selected some text. $text"] == "yes"} {
  868.         set default [getSelect]
  869.     } 
  870.     }
  871.     if {$default == ""} {
  872.     set default [getline "Enter template text (you can edit it later)" $default]
  873.     }
  874.     if {$var != ""} {
  875.     return [elec::_MakeIntoInsertion $default $var]
  876.     } else {
  877.     return $default
  878.     }
  879. }
  880.  
  881. proc global::_editDocumentTemplate {{def ""}} {
  882.     global DocprojmodeVars
  883.     if {$def == ""} {
  884.     set title "Create a new document template" 
  885.     set def {"" "" "By File Extension" "t_XXX" $DocprojmodeVars(currentProject) ""}
  886.     set new 1
  887.     } else {
  888.     set title "Edit document template" 
  889.     set new 0
  890.     }
  891.     
  892.     global docProject
  893.     set name ""
  894.     while { $name == ""} {
  895.     set y 40
  896.     set yb 220
  897.     set res [eval dialog -w 380 -h 340 \
  898.       [dialog::title $title 380] \
  899.       [dialog::button "OK" 290 yb] \
  900.       [dialog::button "Cancel" 290 yb] \
  901.       [dialog::textedit "Descriptive Name" [lindex $def 1] 10 y 15] \
  902.       [dialog::textedit "Modes (blank = all)" [lindex $def 0] 10 y 15] \
  903.       [dialog::textedit "Procedure name" [lindex $def 3] 10 y 15] \
  904.       [dialog::text "Descriptive header for this document template" 10 y] \
  905.       [dialog::text "(if 'Source', or 'Header', the mode must define" 10 y] \
  906.       [dialog::text "headerSuffices and sourceSuffices vars)" 10 y] \
  907.       [dialog::menu 10 y [list "None" "-" "Basic" "Source" "Header" "Either"] [lindex $def 2]] \
  908.       [dialog::text "Project name" 10 y] \
  909.       [dialog::menu 10 y $docProject(name) [lindex $def 4]] \
  910.       [dialog::textedit "List of sub-types" [lindex $def 5] 10 y 30] \
  911.       ]
  912.     if [lindex $res 1] { error "Cancel" } 
  913.     set i 1
  914.     foreach var {name modes procedure filetype proj subproj} {
  915.         set $var [lindex $res [incr i]]
  916.     }
  917.     if {$name == ""} { beep ; message "You must enter a name." }
  918.     }    
  919.     if {$modes == ""} {set modes "*"}
  920.     if {$filetype == "Either"} {set filetype "*"}
  921.     if {$proj == "None"} {set proj "*"}
  922.     if {$procedure == ""} {set procedure "\#"}
  923.     return [list $modes $name $filetype $procedure $proj $subproj]
  924.     
  925. }
  926.  
  927. proc global::editDocumentTemplate {} {
  928.     global modifiedVars elec::DocTemplates
  929.     set tlist [file::docTemplates] 
  930.     set l [listpick -p "Which document template do you want to edit?" $tlist]
  931.     set lind [file::docTemplateIndex $l]
  932.     set l [global::_editDocumentTemplate [file::docTemplateInfo $l]]
  933.     set elec::DocTemplates [lreplace ${elec::DocTemplates} $lind $lind $l]
  934.     lappend modifiedVars elec::DocTemplates
  935. }
  936.  
  937. proc global::removeDocumentTemplate {} {
  938.     global modifiedVars elec::DocTemplates
  939.     set tlist [file::docTemplates] 
  940.     set l [listpick -p "Which document template shall I permanently remove?" $tlist]
  941.     set l [file::docTemplateIndex $l]
  942.     set elec::DocTemplates [lreplace ${elec::DocTemplates} $l $l]
  943.     lappend modifiedVars elec::DocTemplates
  944. }
  945.  
  946. ## Create this sort of stuff.
  947.  # set docProject(name) [list    "None" "EvoX" "Vince's Additions" "Cpptcl"]
  948.  # set docProject(addendum) {    {none} {evolution in complex systems} \
  949.  #       {an extension package for Alpha}    {connecting    C++    with Tcl} }
  950.  # set docProject(default_modes) { {}    {C C++}    {Tcl} {C C++ Tcl}}
  951.  ##
  952. proc global::newProject {} {
  953.     global docProject
  954.     set res [global::_editProject]
  955.     set i -1
  956.     foreach var {name addendum license extra default_modes} {
  957.     lappend docProject($var) [lindex $res [incr i]]
  958.     }
  959.     global modifiedArrVars
  960.     lappend modifiedArrVars docProject
  961.     addMenuItem -m {Current Project} [lindex $res 0]
  962.     Docproj::changeProject [lindex $res 0]
  963. }
  964. proc global::_editProject {{def ""}} {
  965.     if {$def == ""} {
  966.     set title "Create a new project"
  967.     set def [list "Vince's Additions" \
  968.       "an extension package for Alpha" "seeFileLicenseTerms" \
  969.       "See the file \"license.terms\" for information on usage and redistribution of this file, and for a DISCLAIMER OF ALL WARRANTIES." ""]
  970.     } else {
  971.     set title "Edit a project"
  972.     }
  973.     set y 40
  974.     set yb 270
  975.     global elec::LicenseTemplates
  976.     set res [eval dialog -w 380 -h 325 \
  977.       [dialog::title $title 360] \
  978.       [dialog::button "OK" 290 yb] \
  979.       [dialog::button "Cancel" 290 yb] \
  980.       [dialog::textedit "Short Descriptive Name" [lindex $def 0] 10 y 15] \
  981.       [dialog::textedit "Longer Description to append to the above" [lindex $def 1] 10 y 25] \
  982.       [dialog::text "License type for header comments" 10 y] \
  983.       [dialog::menu 10 y ${elec::LicenseTemplates} [lindex $def 2]] \
  984.       [dialog::textedit "Additional text for end of header comments" [lindex $def 3] 10 y 35 5] \
  985.       [dialog::textedit "Modes (blank = all)" [lindex $def 4] 10 y 15] \
  986.       ]
  987.     if [lindex $res 1] { error "Cancel" }
  988.     return [lrange $res 2 6]    
  989. }
  990.  
  991. proc global::editProject {} {
  992.     global docProject modifiedArrVars
  993.     set l [listpick -p "Which project do you wish to edit?" \
  994.       -L [file::projectName] $docProject(name)]
  995.     set item [lsearch -exact $docProject(name) $l]
  996.     foreach uvar {name addendum license extra default_modes} {
  997.     lappend def [lindex $docProject($uvar) $item]
  998.     }
  999.     set def [global::_editProject $def]
  1000.     set i -1
  1001.     foreach uvar {name addendum license extra default_modes} {
  1002.     set docProject($uvar) [lreplace $docProject($uvar) $item $item [lindex $def [incr i]]]
  1003.     }
  1004.     lappend modifiedArrVars docProject
  1005. }
  1006.  
  1007. proc global::removeProject {} {
  1008.     global docProject modifiedArrVars
  1009.     set l [listpick -p "Which project shall I permanently remove?" $docProject(name)]
  1010.     set item [lsearch -exact $docProject(name) $l]
  1011.     foreach uvar {name addendum license extra default_modes} {
  1012.     set docProject($uvar) [lreplace $docProject($uvar) $item $item]
  1013.     }
  1014.     lappend modifiedArrVars docProject
  1015.     if {[file::projectName] == $l} {
  1016.     Docproj::changeProject "None"
  1017.     }
  1018.     deleteMenuItem -m {Current Project} $l
  1019. }
  1020.  
  1021.  
  1022.  
  1023.  
  1024.  
  1025.  
  1026.  
  1027.  
  1028.  
  1029.  
  1030.